**Microprocessor System Design**

**ECE – 485/585**

**Fall 2018**

**Final Project – Cache Simulation**

Vinitha Baddam <[vbaddam@pdx.edu](mailto:vbaddam@pdx.edu)>

Michael Bourquin <[bourquin@pdx.edu](mailto:bourquin@pdx.edu)>

Hima Ethakota <hima@pdx.edu>

# Contents

1. OVERVIEW 2. SPECIFICATIONS 3. ASSUMPTIONS 4. SOURCE CODE 5. TEST CASES WITH OUTPUT

# **1. OVERVIEW:**

This project requires the simulation of a split L1 cache for a 32-bit processor with having multiple processors. The data cache is 8-way set associative having 16k sets and 64-byte lines and the instruction cache is 4-way set associative having 16k sets and 64-byte lines. The cache employs MESI protocol to maintain cache coherence. The L1 data cache is write-back using write allocate and is write-back except for the first write to a line which is write-through. Both caches employ LRU replacement policy and are backed by a shared L2 cache. The number of reads, writes, hits, misses and the hit rate are calculated and recorded.

![](data:image/png;base64,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)

# **2. SPECIFICATIONS**

* The 32-bit address is broken into the offset bits, index bits and tag bits. The data cache is 8-way set associative having 16K sets and 64-byte lines and the instruction cache is a 4-way set associative having 16K sets and 64-byte lines.

**31:20 = 12-bit tag**

**19:6 = 14-bit index**

**5:0 = 6-bit offset**

* We give a command to perform various functions in L1 and L2 cache as the project specifies. The output is simulated and the cache contents are displayed in the report/statistics. There is a command for print which will display the contents in the cache that comprise of offset bits, index bits, tag bits, MESI and LRU bits. For the data cache as it is 8-way associative, we have 4 bits for LRU and for the instruction cache being 4-way associative, we have 2 bits for the LRU. MESI bits are represented as 2 bits.
* When printing the contents and state of the cache in response to a 9 in the trace file, we show only the valid lines in the cache along with way and appropriate state and LRU bits.
* Following is the trace file format

**n <address>**

Where n is

0 - Read data request to L1 data cache

1 - Write data request to L1 data cache

2 - Instruction fetch (a read request to L1 instruction cache)

3 - Invalidate command from L2

4 - Data request from L2 (in response to snoop)

8 - Clear the cache and reset all state (and statistics)

9 - Print contents and state of the cache (allow subsequent trace activity)

And the address will be a hex value.

There will be no address for commands (n) 8 and 9

* In order to maintain inclusivity and implement the MESI protocol the L1 caches may have to communicate with the shared L2 cache. To simulate this, you should display the following messages (where <address> is a hexadecimal address).
* Return data to L2 <address>

In response to a 4 in the trace file your cache should signal that it’s returning the data for that line (if present and modified)

* Write to L2 <address>

This operation is used to write back a modified line to L2 upon eviction from the L1 cache. It is also used for an initial write through when a cache line is written for the first time so that the L2 knows it’s been modified and has the correct data

* Read from L2 <address>

This operation is used to obtain the data from L2 on an L1 cache miss

* Read for Ownership from L2 <address>

This operation is used to obtain the data from L2 on an L1 cache write miss

* Maintain and report the following key statistics of cache usage for each cache and display them upon completion of execution of each trace:
* Number of cache reads
* Number of cache writes
* Number of cache hits
* Number of cache misses
* Cache hit ratio

# **3. ASSUMPTIONS**

* In case of designing the L1 cache, the following assumptions were made:
* The cache employs MESI protocol and also follow inclusivity hierarchy to maintain cache coherence. We use 2 bits to represent the MESI bits in the caches.
* LRU replacement policy is used to update the cache after every transaction takes place. The LRU policy we use is the counter method.
* There is no data hence the cache contents would display the LRU, MESI, tag, index and offset bits.
* The data cache is a write through cache in the first transaction and from the next transaction is a write back one.
* All read and write operations are referred to single byte locations.
* A read request from L2 cache can be because of i) Read miss in other processor’s cache ii) Write miss in other processor’s cache, hence we assume that the processors employ write allocate.
* The MESI bits which are of 2-bit size are encoded as:
* Invalid = 2'b00
* Exclusive = 2'b01
* Shared = 2'b10
* Modified = 2'b11
* The LRU replacement policy is done using the counter method and the most recently used bit is encoded as 111 for data cache (11 for instruction cache) and the least recently used bit is encoded as 000 for data cache (00 for instruction cache). The bits higher than the most recently used bits are decremented by 1 and then updated accordingly.
* In our simulation we have considered two mode output.
* MODE=0: In this mode, our simulation displays only the required summary of usage statistics and responses to 9s in the trace file and nothing else
* MODE=1: In this mode, our simulation should display everything from mode 0 but also display the communication messages to the L2 as described in the specification and nothing else.

# **4. SOURCE CODE:**

The Verilog source code for our project is reproduced below:

**4.1 tb\_CacheSim.v**

*/\**

*// ECE 485/585: Microprocessor System Design*

*// Final Project*

*// Fall 2018*

*// File : tb\_CacheSim.v (Test Bench)*

*// Authors : Vinitha Baddam, Michael Bourquin and Hima Ethakota*

*// Description : This module reads a stimulus file*

\*/

**`define** AddressBits 32 *//Addressbits*

*//+TRACE=latest.txt +MODE=0*

**module** tb\_CacheSim();

*//clock parameters*

**parameter** CLOCK\_CYCLE = 20;

**parameter** CLOCK\_WIDTH = CLOCK\_CYCLE/2;

**parameter** TRUE = 1'b1;

**parameter** FALSE = 1'b0;

**parameter** RESET = 4'd8; // for reset command

**reg** clk; //clock

**integer** fptr; // the file handle

**reg** done; // trace fle processing status

**reg** [3:0] command; // command n from trace file

**reg** [`AddressBits-1:0] address; // address from trace file

**reg** [8\*100:0] filename; //string trace file name

**reg** mode; //output mode

**integer** result;

**CACHE\_SIMULATION** cache\_sim(

.clk(clk),

.command(command),

.address(address),

.mode(mode),

.done(done)

);

**initial**

**begin**

clk = FALSE;

done = FALSE;

*// Check to make sure that a TRACE file was provided*

**if**($value$plusargs("TRACE=%s", filename) == FALSE)

**begin**

**$display**("Please enter a valid trace file name on plusargs");

**$finish;**

**end**

*// If it was, open the file*

fptr = **$fopen**(filename , "r");

**if**($value$plusargs("MODE=%d", mode) == FALSE)

**begin**

**$display**("Please enter a valid mode on plusargs!");

**$finish**;

**end**

*// simulate initial reset*

#CLOCK\_WIDTH clk = FALSE;

command = RESET; *//for reset*

address = 32'b0;

#CLOCK\_WIDTH clk = TRUE;

*// While there are lines left to be read :*

**while**(!$feof(fptr))

**begin**

*// Parse the line*

#CLOCK\_WIDTH clk = FALSE;

result = $fscanf(fptr,"%d", command);

*//check if the command is 8 or 9*

**if**(command != 8 && command != 9)

**begin**

result = $fscanf(fptr,"%h", address);

**end**

#CLOCK\_WIDTH clk = TRUE;

**end**

*// Close the file, and finish up*

**$fclose**(fptr);

#CLOCK\_WIDTH clk = FALSE;

done = TRUE; *// set done to true to print statistics*

#CLOCK\_WIDTH clk = TRUE;

**$stop;**

**end**

**endmodule**

**4.2 CacheSimulation.v**

*/\**

*// ECE 485/585: Microprocessor System Design*

*// Final Project*

*// Fall 2018*

*// File : CacheSimulation.v*

*// Authors : Vinitha Baddam, Michael Bourquin and Hima Ethakota*

*// Description : This module makes a call to all other modules*

\*/

**`define** AddressBits 32 *//Addressbits*

**module** CACHE\_SIMULATION(

**input** clk,

**input** [3:0] command,

*//input address*,

**input** [`AddressBits-1:0] address,

**input** mode,

**input** done

);

*//valid commands from trace file*

**parameter** READ = 4'd0;

**parameter** WRITE = 4'd1;

**parameter** INSTRUCTION\_FETCH = 4'd2;

**parameter** INVALIDATE = 4'd3;

**parameter** SNOOP = 4'd4;

**parameter** RESET = 4'd8;

**parameter** PRINT = 4'd9;

*//signals for statistics*

**wire** [31:0]

d\_read\_hit,

d\_read\_miss,

d\_reads,

d\_write\_hit,

d\_write\_miss,

d\_writes,

i\_hit,

i\_miss,

i\_reads;

*//To call data cache*

**DATA\_CACHE** d\_cache(

.clk(clk),

.command(command),

.address(address),

.mode(mode),

.DC\_Read\_Hit(d\_read\_hit),

.DC\_Read\_Miss(d\_read\_miss),

.DC\_Reads(d\_reads),

.DC\_Write\_Hit(d\_write\_hit),

.DC\_Write\_Miss(d\_write\_miss),

.DC\_Writes(d\_writes)

);

*//To call instruction cache*

**INSTRUCTION\_CACHE** i\_cache(

.clk(clk),

.command(command),

.address(address),

.mode(mode),

.IC\_Read\_Hit(i\_hit),

.IC\_Read\_Miss(i\_miss),

.IC\_Reads(i\_reads)

);

*//To print statistics on done status*

**STATISTICS** stats(

.done(done),

.DC\_Read\_Hit(d\_read\_hit),

.DC\_Read\_Miss(d\_read\_miss),

.DC\_Reads(d\_reads),

.DC\_Write\_Hit(d\_write\_hit),

.DC\_Write\_Miss(d\_write\_miss),

.DC\_Writes(d\_writes),

.IC\_Read\_Hit(i\_hit),

.IC\_Read\_Miss(i\_miss),

.IC\_Reads(i\_reads)

);

**endmodule**

**4.3 InstructionCache.v**

***/\****

*// ECE 485/585: Microprocessor System Design*

*// Final Project*

*// Fall 2018*

*// File : InstructionCache.v*

*// Authors : Vinitha Baddam, Michael Bourquin and Hima Ethakota*

*// Description : This module is for all the instruction cache operations*

*\*/*

**`defin**e InstructionCacheWay 4 *//InstructionCacheWay*

**`define** InstructionCacheSet 16\*1024 *//InstructionCacheset*

**`define** CacheLineSize 64  *//linelength*

**`define** AddressBits 32 *//AddressBits*

**module** INSTRUCTION\_CACHE(

**input** clk,

**input** [3:0] command,

**input** [`AddressBits-1:0] address,

**input** mode,

**output reg** [31:0] IC\_Read\_Hit = 32'b0,

**output reg** [31:0] IC\_Read\_Miss = 32'b0,

**output reg** [31:0] IC\_Reads = 32'b0

);

*//MESI protocal*

**parameter**

Invalid = 2'b00,

Exclusive = 2'b01,

Shared = 2'b10,

Modified = 2'b11;

*//valid commands from trace file*

**parameter**

//READ = 4'd0,

//WRITE = 4'd1,

INSTRUCTION\_FETCH = 4'd2,

INVALIDATE = 4'd3,

//SNOOP = 4'd4,

RESET = 4'd8,

PRINT = 4'd9;

*//Instruction cache index, offset, tag and LRU bits*

**parameter**

IC\_IndexBits = $clog2(`InstructionCacheSet),

IC\_OffsetBits = $clog2(`CacheLineSize),

IC\_TagBits = `AddressBits -(IC\_OffsetBits+IC\_IndexBits),

IC\_LRUBits = $clog2(`InstructionCacheWay),

MESI\_Size = 2;

*//InstructionCache*

**reg** [IC\_TagBits + IC\_LRUBits + MESI\_Size-1:0] InstructionCache[0:`InstructionCacheWay-1] [0:`InstructionCacheSet-1];

*//Cache Instruction Line*

**reg** [IC\_TagBits + IC\_LRUBits + MESI\_Size-1:0] InstructionLine;

**reg** [1:0] MESI\_Bits;

**reg** [IC\_OffsetBits-1:0] Offset;

**reg** [IC\_IndexBits-1:0] Index;

**reg** [IC\_TagBits-1:0] Tag;

*//for instruction cache*

**reg** [IC\_OffsetBits-1:0] IC\_Offset;

**reg** [IC\_IndexBits-1:0] IC\_Index;

**reg** [IC\_TagBits-1:0] IC\_Tag;

**reg** [IC\_LRUBits-1:0] IC\_LRU;

*//These are used to print the cache content*

**reg** [IC\_TagBits-1:0] print\_Tag [0:`InstructionCacheWay];

**reg** [IC\_LRUBits-1:0] print\_LRU [0:`InstructionCacheWay];

**reg** [1:0] print\_MESI [0:`InstructionCacheWay];

**parameter** TRUE = 1'b1;

**parameter** FALSE = 1'b0;

**integer** way, found, replaced, w, s, temp;

**reg** print;

*//Execute below on clock positive edge*

**always** **@**(**posedge** clk)

**begin**

*//Parsing address to get offset, index and tag bits*

Offset = address[IC\_OffsetBits-1:0];

Index = address[IC\_OffsetBits+IC\_IndexBits-1:IC\_OffsetBits];

Tag = address[`AddressBits-1:`AddressBits-IC\_TagBits];

*//Initially found and replaced values will be false*

found=0;

replaced=0;

***case***(command)

*//2 instruction fetch (a read request to L1 instruction cache)*

**INSTRUCTION\_FETCH:**

**begin**

IC\_Reads=IC\_Reads+1; *//reads counter*

*//see if the tag of the fetch address matches any tag in those set lines*

**for**(way=0;way<`InstructionCacheWay;way=way+1)

**begin**

InstructionLine = InstructionCache[way][Index];

MESI\_Bits = InstructionLine[IC\_TagBits + IC\_LRUBits + MESI\_Size-1: IC\_TagBits+ IC\_LRUBits];

IC\_Tag = InstructionLine[IC\_TagBits-1:0];

*//if valid and tag match*

**if**((MESI\_Bits != Invalid) && (IC\_Tag == Tag))

**begin**

IC\_Read\_Hit=IC\_Read\_Hit+1; *//read hit counter*

found=1; *//data in cache found!*

IC\_LRU = InstructionLine[IC\_TagBits+IC\_LRUBits-1: IC\_TagBits];

InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits +IC\_LRUBits] = Shared;

InstructionCache[way][Index] = InstructionLine;

**end**

**end**

*// If tag is not found in the cache set from above*

**if**(found==0)

**begin**

IC\_Read\_Miss=IC\_Read\_Miss+1; *//write miss counter*

**if**(mode == 1)

**$display**("Read from L2 %h", address); *//add lru and write*

*//find an invalid way in set and put tag bits and set lru 111 and decreament lru for other ways by 1*

**for**(way=0;way<`InstructionCacheWay;way=way+1)

**begin**

InstructionLine=InstructionCache[way][Index];

MESI\_Bits=InstructionLine[IC\_TagBits+IC\_LRUBits+ MESI\_Size-1:IC\_TagBits+ IC\_LRUBits];

**if** ((MESI\_Bits == Invalid) && (found!=1))

**begin**

found=1; *// invalid way found*

IC\_LRU=InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits];

InstructionLine[IC\_TagBits-1:0] = Tag;

InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits+ IC\_LRUBits] = Exclusive;

InstructionCache[way][Index] = InstructionLine;

**end**

**end**

*// if no invalid way, look for LRU=000 and check mesi bits*

**if**(found!=1)

**begin**

**for**(way=0;way<`InstructionCacheWay;way=way+1)

**begin**

InstructionLine=InstructionCache[way][Index];

MESI\_Bits =InstructionLine[IC\_TagBits+IC\_LRUBits+ MESI\_Size-1:IC\_TagBits + IC\_LRUBits];

IC\_LRU=InstructionLine[IC\_TagBits+IC\_LRUBits-1: IC\_TagBits];

**if** (IC\_LRU == 0 && replaced == 0)

**begin**

replaced = 1;

*// if mesi bit modified write to L2 and replace,ie,* put tag bits there

**if**(MESI\_Bits == Modified)

**begin**

Offset = 0;

**if**(mode == 1)

$**display**("Write to L2 %h", {InstructionLine[IC\_TagBits-1:0], Index, Offset});

**end**

IC\_LRU=InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits];

InstructionLine[IC\_TagBits-1:0] = Tag;

InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits+ IC\_LRUBits] = Exclusive;

InstructionCache[way][Index] = InstructionLine;

**end**

**end**

IC\_LRU = 0; *//To replace '0'th LRU element*

**end**

**end**

temp = UpdateInstruction\_LRU(IC\_LRU, Index); *// update lru*

**end**

*//3 invalidate command from L2*

**INVALIDATE:**

**begin**

*// look up for line by going to set n comparing tags in ways*

**for**(way=0;way<`InstructionCacheWay;way=way+1)

**begin**

InstructionLine = InstructionCache[way][Index];

MESI\_Bits = InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits+ IC\_LRUBits];

IC\_Tag = InstructionLine[IC\_TagBits-1:0];

**if**(Tag == IC\_Tag) *//if tag match*

**begin**

MESI\_Bits = Invalid; //if line found then set MESI\_Bits as invalid

IC\_LRU = InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits];

InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits+IC\_LRUBits] = MESI\_Bits;

InstructionLine[IC\_TagBits-1:0] = 12'bx;

InstructionCache[way][Index] = InstructionLine;

**end**

**end**

temp = UpdateInstruction\_LRU(IC\_LRU, Index); *// update lru*

**end**

*//8 clear the cache and reset all state (and statistics)*

**RESET:**

**begin**

**for** (s=0; s<`InstructionCacheSet; s=s+1)

**begin**

**for** (w=0; w<`InstructionCacheWay; w=w+1)

**begin**

InstructionLine=InstructionCache[w][s];

*//set all the cache MESI bits to Invalid*

InstructionLine[IC\_TagBits + IC\_LRUBits + MESI\_Size-1 : IC\_TagBits + IC\_LRUBits] = Invalid;

*//Initially assign LRU bits to each cache line in a set to line number*

InstructionLine[IC\_TagBits + IC\_LRUBits-1 : IC\_TagBits] = w;

*//set tag bits to x*

InstructionLine[IC\_TagBits-1:0] = 12'bx;

InstructionCache[w][s] = InstructionLine;

**end**

**end**

*//set all summary paramentes to '0' on reset*

IC\_Read\_Hit = 32'b0;

IC\_Read\_Miss = 32'b0;

IC\_Reads = 32'b0;

**end**

*//9 print contents and state of the cache (allow subsequent trace activity)*

**PRINT:**

**begin**

**$display**("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_");

**$display**(" ");

**$display**(" INSTRUCTION CACHE CONTENTS ");

**$display**("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_");

**for** (s=0;s<`InstructionCacheSet;s=s+1)

**begin**

print = FALSE;

**for** (w=0;w<`InstructionCacheWay;w=w+1)

**begin**

InstructionLine = InstructionCache[w][s];

MESI\_Bits = InstructionLine[IC\_TagBits+IC\_LRUBits+MESI\_Size-1:IC\_TagBits+ IC\_LRUBits];

IC\_LRU = InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits];

IC\_Tag = InstructionLine[IC\_TagBits-1:0];

**if**(print == TRUE || MESI\_Bits != Invalid || IC\_Tag != "x")

**begin**

**if**(print == FALSE)

**begin**

print = TRUE;

w = -1;

**end**

**else**

**begin**

print\_MESI[w] = MESI\_Bits;

print\_LRU[w] = IC\_LRU;

print\_Tag[w] = IC\_Tag;

**end**

**end**

**end**

**if**(print == TRUE)

**begin**

**$display**("Set Index : %h", s);

**$display**("Way: 1 2 3 4");

**$display**("Tag: %h %h %h %h", print\_Tag[0], print\_Tag[1], print\_Tag[2], print\_Tag[3]);

**$display**("LRU: %b %b %b %b", print\_LRU[0], print\_LRU[1], print\_LRU[2], print\_LRU[3]);

**$display**("MESI: %s %s %s %s", Get\_MESI\_ID(print\_MESI[0]), Get\_MESI\_ID(print\_MESI[1]), Get\_MESI\_ID(print\_MESI[2]), Get\_MESI\_ID(print\_MESI[3]));

**$display**(" \*\* END OF SET \*\* ");

**$display("------------------------------------");**

**end**

**end**

**$display**(" END OF INSTRUCTION CACHE CONTENTS ");

**$display("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_");**

**end**

**endcase**

**end**

*//function to update LRU values of each line in a set based on line reference*

**function** UpdateInstruction\_LRU;

**input** [IC\_LRUBits-1:0] LRU;

**input** [IC\_IndexBits-1:0] index;

**integer** w;

**begin**

*//update the LRU bits of each cache line in a set*

**for** (w=0; w<`InstructionCacheWay; w=w+1)

**begin**

InstructionLine = InstructionCache[w][index];

IC\_LRU = InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits];

*//if it is a most recently used/refered cache line then set the LRU bits to 111*

**if**(IC\_LRU == LRU)

**begin**

InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits] = `InstructionCacheWay-1;

InstructionCache[w][index] = InstructionLine;

**end**

*//LRU bits higher than the most recently used bits are decremented by 1*

**else** if(IC\_LRU > LRU)

**begin**

InstructionLine[IC\_TagBits+IC\_LRUBits-1:IC\_TagBits] = IC\_LRU-1;

InstructionCache[w][index] = InstructionLine;

**end**

**end**

**end**

**endfunction**

*//function to get MESI ids for a given 2 bit MESI value*

**function** [8:0] Get\_MESI\_ID;

**input** [1:0] MESI;

**begin**

**case**(MESI)

2'b00: Get\_MESI\_ID = "I";

2'b01: Get\_MESI\_ID = "E";

2'b10: Get\_MESI\_ID = "S";

2'b11: Get\_MESI\_ID = "M";

**endcase**

**end**

**endfunction**

**endmodule**

**4.4 DataCache.v**

*/\**

*// ECE 485/585: Microprocessor System Design*

*// Final Project*

*// Fall 2018*

*// File : DataCache.v*

*// Authors : Vinitha Baddam, Michael Bourquin and Hima Ethakota*

*// Description : This module is for all the data cache operations*

*\*/*

**`define** DataCacheWay 8 *//DataCacheway*

**`define** DataCacheSet 16\*1024 *//DataCacheset*

**`define** CacheLineSize 64 *//linelength*

**`define** AddressBits 32 *//AddressBits*

**module** DATA\_CACHE(

**input** clk,

**input** [3:0] command,

**input** [`AddressBits-1:0] address,

**input** mode,

**output** **reg** [31:0] DC\_Read\_Hit = 32'b0,

**output** **reg** [31:0] DC\_Read\_Miss = 32'b0,

**output reg** [31:0] DC\_Reads = 32'b0,

**output reg** [31:0] DC\_Write\_Hit = 32'b0,

**output reg** [31:0] DC\_Write\_Miss = 32'b0,

**output reg** [31:0] DC\_Writes = 32'b0

);

*//MESI protocal*

*parameter*

Invalid = 2'b00,

Exclusive = 2'b01,

Shared = 2'b10,

Modified = 2'b11;

*//valid commands from trace file*

**parameter**

READ = 4'd0,

WRITE = 4'd1,

//INSTRUCTION\_FETCH = 4'd2,

INVALIDATE = 4'd3,

SNOOP = 4'd4,

RESET = 4'd8,

PRINT = 4'd9;

*//Data cache index, offset, tag and LRU bits*

**parameter**

DC\_IndexBits = $clog2(`DataCacheSet),

DC\_OffsetBits = $clog2(`CacheLineSize),

DC\_TagBits = `AddressBits -(DC\_OffsetBits+DC\_IndexBits),

DC\_LRUBits = $clog2(`DataCacheWay),

MESI\_Size = 2;

*//Data Cache*

**reg** [DC\_TagBits + DC\_LRUBits + MESI\_Size-1:0] DataCache[0:`DataCacheWay-1] [0:`DataCacheSet-1];

*//Cache Data Line*

**reg** [DC\_TagBits + DC\_LRUBits + MESI\_Size-1:0] DataLine;

**reg** [1:0] MESI\_Bits;

**reg** [DC\_OffsetBits-1:0] Offset;

**reg** [DC\_IndexBits-1:0] Index;

**reg** [DC\_TagBits-1:0] Tag;

*//for data cache*

**reg** [DC\_OffsetBits-1:0] DC\_Offset;

**reg** [DC\_IndexBits-1:0] DC\_Index;

**reg** [DC\_TagBits-1:0] DC\_Tag;

**reg** [DC\_LRUBits-1:0] DC\_LRU;

*//These are used to print the cache content*

**reg** [DC\_TagBits-1:0] print\_Tag [0:`DataCacheWay];

**reg** [DC\_LRUBits-1:0] print\_LRU [0:`DataCacheWay];

**reg** [1:0] print\_MESI [0:`DataCacheWay];

**parameter** TRUE = 1'b1;

**parameter** FALSE = 1'b0;

**integer** way, found, replaced, w, s, temp;

**reg** print;

*//Execute below on clock positive edge*

**always** **@**(**posedge** clk)

**begin**

*//Parsing address to get offset, index and tag bits*

Offset = address[DC\_OffsetBits-1:0];

Index = address[DC\_OffsetBits+DC\_IndexBits-1:DC\_OffsetBits];

Tag = address[`AddressBits-1:`AddressBits-DC\_TagBits];

*//Initially found and replaced values will be false*

found=0;

replaced=0;

**case**(command)

*//0 read data request to L1 data cache*

**READ**:

**begin**

DC\_Reads=DC\_Reads+1*; //reads counter*

*//see if the tag of the read address matches any tag in those set lines*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_Tag = DataLine[DC\_TagBits-1:0];

*//if valid and tag match*

**if**((MESI\_Bits != Invalid) && (DC\_Tag == Tag))

**begin**

DC\_Read\_Hit=DC\_Read\_Hit+1; *//read hit counter*

DC\_LRU=DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

found=1; *//data in cache found!*

**if** (MESI\_Bits == Exclusive)

**begin**

MESI\_Bits = Shared; *//change the MESI bits to Shared if it was Exclusive*

**end**

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = MESI\_Bits;

DataCache[way][Index] = DataLine;

**end**

**end**

*// If tag is not found in the cache set from above*

**if**(found==0)

**begin**

DC\_Read\_Miss=DC\_Read\_Miss+1; *//read miss counter*

**if**(mode == 1)

**$display**("Read from L2 %h", address); *//add lru and write*

*//find an invalid way in set and put tag bits and set lru 111 and decreament lru for other ways by 1*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

**if** ((MESI\_Bits == Invalid) && (found!=1))

**begin**

found=1; *// invalid way found*

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits-1:0] = Tag;

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = Exclusive;

DataCache[way][Index] = DataLine;

**end**

**end**

*// if no invalid way, look for LRU=000 and check mesi bits*

**if**(found!=1)

**begin**

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1: DC\_TagBits+DC\_LRUBits];

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

**if** (DC\_LRU == 0 && replaced == 0)

**begin**

replaced = 1;

*// if mesi bit modified write to L2 and replace,ie, put tag bits there*

**if**(MESI\_Bits == Modified)

**begin**

Offset = 0;

**if**(mode == 1)

**$display**("Write to L2 %h", {DataLine[DC\_TagBits-1:0], Index, Offset});

**end**

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits-1:0] = Tag;

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits] = Exclusive;

DataCache[way][Index] = DataLine;

**end**

**end**

DC\_LRU = 0; *//To replace '0'th LRU element*

**end**

**end**

temp = UpdateData\_LRU(DC\_LRU, Index);

**end**

*//1 write data request to L1 data cache*

**WRITE:**

**begin**

DC\_Writes=DC\_Writes+1; *//writes counter*

*//see if the tag of the write address matches any tag in those set lines*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_Tag = DataLine[DC\_TagBits-1:0];

*//if valid and tag match*

**if**((MESI\_Bits != Invalid) && (DC\_Tag == Tag))

**begin**

DC\_Write\_Hit=DC\_Write\_Hit+1; *//write hit counter*

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

found=1; // cache line to be written found!

**if**(MESI\_Bits == Shared)

**begin**

**if**(mode == 1)

**$display**("Write to L2 %h", address);

MESI\_Bits = Exclusive;

**end**

**else if** (MESI\_Bits == Exclusive)

**begin**

MESI\_Bits = Modified;

**end**

**else if** (MESI\_Bits == Modified)

**begin**

MESI\_Bits = Modified;

**end**

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = MESI\_Bits;

DataCache[way][Index] = DataLine;

**end**

**end**

*// If tag is not found in the cache set from above*

**if**(found==0)

**begin**

DC\_Write\_Miss=DC\_Write\_Miss+1; *//write miss counter*

**if**(mode == 1)

**$display**("Read for Ownership from L2 %h", address); *//add lru and write*

*//find an invalid way in set and put tag bits and set lru 111 and decreament lru for other ways by 1*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine=DataCache[way][Index];

MESI\_Bits=DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

**if**((MESI\_Bits == Invalid) && (found!=1))

**begin**

found=1; *// invalid way found*

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits-1:0] = Tag;

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = Exclusive;

DataCache[way][Index] = DataLine;

*// <write data to this line>*

*//first write is write through*

**if**(mode == 1)

**$display**("Write to L2 %h ",address);

**end**

**end**

*// if no invalid way, look for LRU=000 and check mesi bits*

**if**(found!=1)

**begin**

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

**if** (DC\_LRU == 0 && replaced == 0)

**begin**

replaced = 1;

*// if mesi bit modified write to L2 and replace,ie, put tag bits there*

**if**(MESI\_Bits == Modified)

**begin**

Offset = 0;

**if**(mode == 1)

**$display**("Write to L2 %h", {DataLine[DC\_TagBits-1:0], Index, Offset});

**end**

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits-1:0] = Tag;

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits] = Modified;

DataCache[way][Index] = DataLine;

**end**

**end**

DC\_LRU = 0; *//To replace '0'th LRU element*

**end**

**end**

temp = UpdateData\_LRU(DC\_LRU, Index); *// update lru*

**end**

*//3 invalidate command from L2*

**INVALIDATE:**

**begin**

*// look up for line by going to set and comparing tags in ways*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_Tag = DataLine[DC\_TagBits-1:0];

**if**(Tag == DC\_Tag) //if tag match

**begin**

MESI\_Bits = Invalid; *//if line found then set MESI\_Bits as invalid*

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = MESI\_Bits;

DataLine[DC\_TagBits-1:0] = 12'bx;

DataCache[way][Index] = DataLine;

**end**

**end**

temp = UpdateData\_LRU(DC\_LRU, Index); *// update lru*

**end**

*//4 data request from L2 (in response to snoop)*

**SNOOP:**

**begin**

*//look up for line with mesi modified*

**for**(way=0;way<`DataCacheWay;way=way+1)

**begin**

DataLine = DataCache[way][Index];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_Tag = DataLine[DC\_TagBits-1:0];

**if**((MESI\_Bits == Modified) && (Tag == DC\_Tag)) //if valid and tag match

**begin**

found=1; *// modified cache line found!*

**if**(mode == 1)

**$display**("Return data to L2 %h",address); *//Return data to L2 <address>*

**end**

**if**(Tag == DC\_Tag)

**begin**

found=1; *// modified cache line found!*

MESI\_Bits = Invalid; //if found then change to invalid

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+DC\_LRUBits] = MESI\_Bits;

DataLine[DC\_TagBits-1:0] = 12'bx;

DataCache[way][Index] = DataLine;

**end**

**end**

temp = UpdateData\_LRU(DC\_LRU, Index); *// update lru*

**end**

*//8 clear the cache and reset all state (and statistics)*

**RESET:**

**begin**

**for** (s=0; s<`DataCacheSet; s=s+1)

**begin**

**for** (w=0; w<`DataCacheWay; w=w+1)

**begin**

DataLine=DataCache[w][s];

*//set all the cache MESI bits to Invalid*

DataLine[DC\_TagBits + DC\_LRUBits + MESI\_Size-1 : DC\_TagBits + DC\_LRUBits] = Invalid;

*//Initially assign LRU bits to each cache line in a set to line number*

DataLine[DC\_TagBits + DC\_LRUBits-1 : DC\_TagBits] = w;

*//set tag bits to x*

DataLine[DC\_TagBits-1:0] = 12'bx;

DataCache[w][s] = DataLine;

**end**

**end**

*//set all summary paramentes to '0' on reset*

DC\_Read\_Hit = 32'b0;

DC\_Read\_Miss = 32'b0;

DC\_Reads = 32'b0;

DC\_Write\_Hit = 32'b0;

DC\_Write\_Miss = 32'b0;

DC\_Writes = 32'b0;

**end**

*//9 print contents and state of the cache (allow subsequent trace activity)*

**PRINT:**

**begin**

**$display**("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_ \_\_\_\_\_\_\_\_");

**$display**(" ");

**$display**(" DATA CACHE CONTENTS ");

**$display**("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_ \_\_\_\_\_\_");

**for** (s=0;s<`DataCacheSet;s=s+1)

**begin**

print = FALSE;

**for** (w=0;w<`DataCacheWay;w=w+1)

**begin**

DataLine = DataCache[w][s];

MESI\_Bits = DataLine[DC\_TagBits+DC\_LRUBits+MESI\_Size-1:DC\_TagBits+ DC\_LRUBits];

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

DC\_Tag = DataLine[DC\_TagBits-1:0];

**if**(print == TRUE || MESI\_Bits != Invalid || DC\_Tag != "x")

**begin**

**if**(print == FALSE)

**begin**

print = TRUE;

w = -1;

**end**

**else**

**begin**

print\_MESI[w] = MESI\_Bits;

print\_LRU[w] = DC\_LRU;

print\_Tag[w] = DC\_Tag;

**end**

**end**

**end**

**if**(print == TRUE)

**begin**

**$display**("Set Index : %h", s);

**$display**("Way: 1 2 3 4 5 6 7 8");

**$display**("Tag: %h %h %h %h %h %h %h %h", print\_Tag[0], print\_Tag[1], print\_Tag[2], print\_Tag[3], print\_Tag[4], print\_Tag[5], print\_Tag[6], print\_Tag[7]);

**$display**("LRU: %b %b %b %b %b %b %b %b", print\_LRU[0], print\_LRU[1], print\_LRU[2], print\_LRU[3], print\_LRU[4], print\_LRU[5], print\_LRU[6], print\_LRU[7]);

**$display**("MESI: %s %s %s %s %s %s %s %s", Get\_MESI\_ID(print\_MESI[0]), Get\_MESI\_ID(print\_MESI[1]), Get\_MESI\_ID(print\_MESI[2]), Get\_MESI\_ID(print\_MESI[3]), Get\_MESI\_ID(print\_MESI[4]), Get\_MESI\_ID(print\_MESI[5]), Get\_MESI\_ID(print\_MESI[6]), Get\_MESI\_ID(print\_MESI[7]));

**$display**(" \*\* END OF SET \*\* ");

**$display**("------------------------------------------------------------------");

**end**

**end**

**$display**(" END OF DATA CACHE CONTENTS ");

**$display**("\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_ \_\_\_\_");

**end**

**endcase**

**end**

*//function to update LRU values of each line in a set based on line reference*

**function** UpdateData\_LRU;

**input** [DC\_LRUBits-1:0] LRU;

**input** [DC\_IndexBits-1:0] index;

**integer** w;

**begin**

*//update the LRU bits of each cache line in a set*

**for** (w=0; w<`DataCacheWay; w=w+1)

**begin**

DataLine = DataCache[w][index];

DC\_LRU = DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits];

*//if it is a most recently used/refered cache line then set the LRU bits to 111*

**if**(DC\_LRU == LRU)

**begin**

DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits] = `DataCacheWay-1;

DataCache[w][index] = DataLine;

**end**

*//LRU bits higher than the most recently used bits are decremented by 1*

**else if**(DC\_LRU > LRU)

**begin**

DataLine[DC\_TagBits+DC\_LRUBits-1:DC\_TagBits] = DC\_LRU-1;

DataCache[w][index] = DataLine;

**end**

**end**

**end**

**endfunction**

*//function to get MESI ids for a given 2 bit MESI value*

**function** [8:0] Get\_MESI\_ID;

**input** [1:0] MESI;

**begin**

**case**(MESI)

2'b00: Get\_MESI\_ID = "I";

2'b01: Get\_MESI\_ID = "E";

2'b10: Get\_MESI\_ID = "S";

2'b11: Get\_MESI\_ID = "M";

**endcase**

**4.5 Statistics.v**

*/\**

*// ECE 485/585: Microprocessor System Design*

*// Final Project*

*// Fall 2018*

*// File : Statistics.v*

*// Authors : Vinitha Baddam, Michael Bourquin and Hima Ethakota*

*// Description : This module is for printing statistics*

*\*/*

**module** STATISTICS(

**input** done,

**input** [31:0] DC\_Read\_Hit,

**input** [31:0] DC\_Read\_Miss,

**input** [31:0] DC\_Reads,

**input** [31:0] DC\_Write\_Hit,

**input** [31:0] DC\_Write\_Miss,

**input** [31:0] DC\_Writes,

**input** [31:0] IC\_Read\_Hit,

**input** [31:0] IC\_Read\_Miss,

**input** [31:0] IC\_Reads

);

*//Execute below when done is true*

**always @*(*posedge** done)

**begin**

**$display**("Data Cache Usage Statistics:");

**$display**("Number of cache reads : %d", DC\_Reads);

**$display**("Number of cache writes : %d", DC\_Writes);

**$display**("Number of cache hits : %d", DC\_Read\_Hit + DC\_Write\_Hit);

**$display**("Number of cache misses : %d", DC\_Read\_Miss + DC\_Write\_Miss);

**$display**("Cache hit ratio : %.2f%% \n", (DC\_Reads + DC\_Writes) != 0 ? 100.00 \* (DC\_Read\_Hit + DC\_Write\_Hit)/(DC\_Reads + DC\_Writes) : 0);

**$display**("Instruction Cache Usage Statistics:");

**$display**("Number of cache reads : %d", IC\_Reads);

**$display**("Number of cache hits : %d", IC\_Read\_Hit);

**$display**("Number of cache misses : %d", IC\_Read\_Miss);

**$display**("Cache hit ratio : %.2f%% \n", IC\_Reads != 0 ? 100.00\*(IC\_Read\_Hit)/(IC\_Reads) : 0);

**end**

**endmodule**

# **5. TEST CASES WITH OUTPUT**

# **5.1 explained.txt**

0 984DE132

0 116DE12F

0 100DE130

0 999DE12E

0 645DE10A

0 846DE107

0 211DE128

0 777DE133

9

0 999DE132

1 116DE123

1 666DE135

1 333DE12C

0 846DE10C

0 777DE136

1 ABCDE128

0 116DE101

1 100DE101

1 AAADE101

1 EDCDE101

4 AAADE101

9

**Output:** MODE=0

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Set Index : 00003784

# Way: 1 2 3 4 5 6 7 8

# Tag: 984 116 100 999 645 846 211 777

# LRU: 000 001 010 011 100 101 110 111

# MESI: E E E E E E E E

# \*\* END OF SET \*\*

# ------------------------------------------------------------------

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Set Index : 00003784

# Way: 1 2 3 4 5 6 7 8

# Tag: edc 116 333 xxx abc 846 100 777

# LRU: 110 100 000 111 011 001 101 010

# MESI: M M M I M S M S

# \*\* END OF SET \*\*

# ------------------------------------------------------------------

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Data Cache Usage Statistics:

# Number of cache reads : 12

# Number of cache writes : 7

# Number of cache hits : 5

# Number of cache misses : 14

# Cache hit ratio : 26.32%

#

# Instruction Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# \*\* Note: $stop : N:/tb\_CacheSim.v(89)

# Time: 480 ns Iteration: 0 Instance: /tb\_CacheSim

# **5.2 reset.txt**

0 984DE132

0 116DE12F

0 100DE130

0 999DE12E

0 645DE10A

0 846DE107

0 211DE128

0 777DE133

9

8

9

**Output:** MODE=0

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Set Index : 00003784

# Way: 1 2 3 4 5 6 7 8

# Tag: 984 116 100 999 645 846 211 777

# LRU: 000 001 010 011 100 101 110 111

# MESI: E E E E E E E E

# \*\* END OF SET \*\*

# ------------------------------------------------------------------

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Data Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache writes : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# Instruction Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# \*\* Note: $stop : N:/tb\_CacheSim.v(89)

# Time: 260 ns Iteration: 0 Instance: /tb\_CacheSim

# **5.3 EmptyFile.txt**

**Output:** MODE=0

# Data Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache writes : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# Instruction Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# \*\* Note: $stop : N:/tb\_CacheSim.v(89)

# Time: 60 ns Iteration: 0 Instance: /tb\_CacheSim

# **5.4 InstructionFetch.txt**

2 984DE132

2 116DE12F

2 100DE130

2 999DE12E

9

2 645DE10A

2 846DE107

2 211DE128

9

**Output:**

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Set Index : 00003784

# Way: 1 2 3 4

# Tag: 984 116 100 999

# LRU: 00 01 10 11

# MESI: E E E E

# \*\* END OF SET \*\*

# ------------------------------------

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# END OF DATA CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

#

# INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Set Index : 00003784

# Way: 1 2 3 4

# Tag: 645 846 211 999

# LRU: 01 10 11 00

# MESI: E E E E

# \*\* END OF SET \*\*

# ------------------------------------

# END OF INSTRUCTION CACHE CONTENTS

# \_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_\_

# Data Cache Usage Statistics:

# Number of cache reads : 0

# Number of cache writes : 0

# Number of cache hits : 0

# Number of cache misses : 0

# Cache hit ratio : 0.00%

#

# Instruction Cache Usage Statistics:

# Number of cache reads : 7

# Number of cache hits : 0

# Number of cache misses : 7

# Cache hit ratio : 0.00%

#

# \*\* Note: $stop : N:/tb\_CacheSim.v(89)

# Time: 220 ns Iteration: 0 Instance: /tb\_CacheSim